// Copyright 2024 RisingWave Labs
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

use std::sync::Arc;

use risingwave_common::util::sort_util::ColumnOrder;
use risingwave_pb::stream_plan::TopNNode;

use super::*;
use crate::common::table::state_table::StateTable;
use crate::executor::{AppendOnlyTopNExecutor, TopNExecutor};

pub struct TopNExecutorBuilder<const APPEND_ONLY: bool>;

impl<const APPEND_ONLY: bool> ExecutorBuilder for TopNExecutorBuilder<APPEND_ONLY> {
    type Node = TopNNode;

    async fn new_boxed_executor(
        params: ExecutorParams,
        node: &Self::Node,
        store: impl StateStore,
    ) -> StreamResult<Executor> {
        let [input]: [_; 1] = params.input.try_into().unwrap();

        let table = node.get_table()?;
        let vnodes = params.vnode_bitmap.map(Arc::new);
        let state_table = StateTable::from_table_catalog(table, store, vnodes).await;
        let storage_key = table
            .get_pk()
            .iter()
            .map(ColumnOrder::from_protobuf)
            .collect();
        let order_by = node
            .order_by
            .iter()
            .map(ColumnOrder::from_protobuf)
            .collect();

        macro_rules! build {
            ($excutor:ident, $with_ties:literal) => {
                Ok($excutor::<_, $with_ties>::new(
                    input,
                    params.actor_context,
                    params.info.schema.clone(),
                    storage_key,
                    (node.offset as usize, node.limit as usize),
                    order_by,
                    state_table,
                )?
                .boxed())
            };
        }

        let exec: StreamResult<Box<dyn Execute>> = match (APPEND_ONLY, node.with_ties) {
            (true, true) => build!(AppendOnlyTopNExecutor, true),
            (true, false) => build!(AppendOnlyTopNExecutor, false),
            (false, true) => build!(TopNExecutor, true),
            (false, false) => build!(TopNExecutor, false),
        };
        Ok((params.info, exec?).into())
    }
}
